package cc.admin.common.util;

import cn.hutool.core.date.DateUnit;
import cn.hutool.core.date.DateUtil;
import com.google.common.collect.Lists;
import org.joda.time.DateTime;
import org.springframework.util.StringUtils;

import java.beans.PropertyEditorSupport;
import java.sql.Timestamp;
import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * 类描述：时间操作定义类
 *
 * @Author: 张代浩
 * @Date:2012-12-8 12:15:03
 * @Version 1.0
 */
public class DateUtils extends PropertyEditorSupport {

    public static ThreadLocal<SimpleDateFormat> date_sdf = new ThreadLocal<SimpleDateFormat>() {
        @Override
        protected SimpleDateFormat initialValue() {
            return new SimpleDateFormat("yyyy-MM-dd");
        }
    };
    public static ThreadLocal<SimpleDateFormat> yyyyMMdd = new ThreadLocal<SimpleDateFormat>() {
        @Override
        protected SimpleDateFormat initialValue() {
            return new SimpleDateFormat("yyyyMMdd");
        }
    };
    public static ThreadLocal<SimpleDateFormat> date_sdf_wz = new ThreadLocal<SimpleDateFormat>() {
        @Override
        protected SimpleDateFormat initialValue() {
            return new SimpleDateFormat("yyyy年MM月dd日");
        }
    };
    public static ThreadLocal<SimpleDateFormat> time_sdf = new ThreadLocal<SimpleDateFormat>() {
        @Override
        protected SimpleDateFormat initialValue() {
            return new SimpleDateFormat("yyyy-MM-dd HH:mm");
        }
    };
    public static ThreadLocal<SimpleDateFormat> yyyymmddhhmmss = new ThreadLocal<SimpleDateFormat>() {
        @Override
        protected SimpleDateFormat initialValue() {
            return new SimpleDateFormat("yyyyMMddHHmmss");
        }
    };
    public static ThreadLocal<SimpleDateFormat> short_time_sdf = new ThreadLocal<SimpleDateFormat>() {
        @Override
        protected SimpleDateFormat initialValue() {
            return new SimpleDateFormat("HH:mm");
        }
    };
    public static ThreadLocal<SimpleDateFormat> datetimeFormat = new ThreadLocal<SimpleDateFormat>() {
        @Override
        protected SimpleDateFormat initialValue() {
            return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        }
    };

    // 以毫秒表示的时间
    private static final long DAY_IN_MILLIS = 24 * 3600 * 1000;
    private static final long HOUR_IN_MILLIS = 3600 * 1000;
    private static final long MINUTE_IN_MILLIS = 60 * 1000;
    private static final long SECOND_IN_MILLIS = 1000;

	public static final String DATE_BASE_REGEX = "\\d{4}-\\d{1,2}-\\d{1,2}";
	public static final String SEPARATOR_REGEX = "(T?|\\s*)";

	public static final String DATE_REGEX = "^\\s*" + DATE_BASE_REGEX + "\\s*$";
	public static final String DATE_H_REGEX = "^\\s*" + DATE_BASE_REGEX + SEPARATOR_REGEX + "\\d\\d\\s*$";
	public static final String DATE_HM_REGEX = "^\\s*" + DATE_BASE_REGEX + SEPARATOR_REGEX + "\\d{1,2}:\\dd{1,2}\\s*$";
	public static final String DATE_HMS_REGEX = "^\\s*" + DATE_BASE_REGEX + SEPARATOR_REGEX + "\\d{1,2}:\\d{1,2}:\\d{1,2}\\s*$";
	public static final String DATE_HMS_M_REGEX = "^\\s*" + DATE_BASE_REGEX + SEPARATOR_REGEX + "\\d{1,2}:\\d{1,2}:\\d{1,2}\\.\\d{1,3}\\s*$";

	/**
	 * 根据开始日期，计算日期所在周
	 *
	 * @param startDate 开始日期
	 * @param calcDate  要计算周的日期
	 * @return
	 */
	public static Integer getWeek(Date startDate, Date calcDate) {
		long days = DateUtil.between(calcDate, startDate, DateUnit.DAY);
		return DateUtil.year(calcDate) * 100 + (int) (days / 7) + 1;
	}

	/**
	 * 根据开始日期，计算日期段所有周
	 *
	 * @param startDate 开始计算日期
	 * @param beginDate 开始时间段
	 * @param endDate   结束时间段
	 * @return
	 */
	public static List<Integer> getWeekList(Date startDate, Date beginDate, Date endDate) {
		List<Integer> resultList = Lists.newArrayList();
		int week = 1;
		for (long i = startDate.getTime(); i < endDate.getTime(); i += 7 * 24 * 60 * 60 * 1000) {
			if (i >= beginDate.getTime()) {
				resultList.add(DateUtil.year(beginDate) * 100 + week);
			}
			week++;
		}
		return resultList;
	}

	/**
	 * 根据开始日期，计算日期段所有周
	 *
	 * @param startDate 开始计算日期
	 * @param beginDate 开始时间段
	 * @param endDate   结束时间段
	 * @return
	 */
	public static List<Integer> getWeekValueList(Date startDate, Date beginDate, Date endDate) {
		List<Integer> resultList = Lists.newArrayList();
		int week = 1;
		for (long i = startDate.getTime(); i < endDate.getTime(); i += 7 * 24 * 60 * 60 * 1000) {
			if (i >= beginDate.getTime()) {
				resultList.add(week);
			}
			week++;
		}
		return resultList;
	}

	/**
	 * 根据开始日期，计算日期段所有周
	 *
	 * @param week 开始计算日期
	 * @return
	 */
	public static List<String> getWeekDate(Date startDate, int week) {
		List<String> resultList = Lists.newArrayList();
		long i = startDate.getTime();
		Long j = (long) 7 * 24 * 60 * 60 * 1000 * (week - 1);
		long day1 = i + j;
		Date beginDate = new Date(day1);
		long x = (long) 7 * 24 * 60 * 60 * 1000 * week;
		long day = i + x;
		Date endDate = new Date(day);
		;
		resultList.add(DateUtils.formatDate(beginDate, "yyyy-MM-dd HH:mm:ss"));
		resultList.add(DateUtils.formatDate(endDate, "yyyy-MM-dd HH:mm:ss"));
		return resultList;
	}

	private enum DATE_FORMAT_ENUM {
		DATE(Pattern.compile(DATE_REGEX), "yyyy-M-d"),
		DATE_H(Pattern.compile(DATE_H_REGEX), "yyyy-M-d H"),
		DATE_HM(Pattern.compile(DATE_HM_REGEX), "yyyy-M-d H:m"),
		DATE_HMS(Pattern.compile(DATE_HMS_REGEX), "yyyy-M-d H:m:s"),
		DATE_HMS_M(Pattern.compile(DATE_HMS_M_REGEX), "yyyy-M-d H:m:s.SSS");

		private Pattern pattern;
		private String format;

		DATE_FORMAT_ENUM(Pattern pattern, String format) {
			this.pattern = pattern;
			this.format = format;
		}

		static Date formatDate(String timeString) throws Exception {
			timeString = prepare(timeString);
			for (DATE_FORMAT_ENUM formatEnum : values()) {
				Matcher matcher = formatEnum.pattern.matcher(timeString);
				if (matcher.find()) {
					SimpleDateFormat sdf = new SimpleDateFormat(formatEnum.format);
					return sdf.parse(timeString);
				}
			}
			throw new Exception("Unparseable date: " + timeString);
		}

		private static String prepare(String timeString) {
			String s = null;
			if (!com.alibaba.druid.util.StringUtils.isEmpty(timeString)) {
				if (timeString.contains("-")) {
					s = timeString;
				} else if (timeString.contains("/")) {
					s = timeString.replaceAll("[/]", "-");
				} else {
					s = timeString.substring(0, 4) + "-" + timeString.substring(4, 6) + "-" + timeString.substring(6);
				}
				int dotStart = s.indexOf(".") + 1;
				boolean hasDot = dotStart > 0;
				int msLength = dotStart + 3;
				int overMsLength = s.length() - msLength;
				int lessMsLength = msLength - s.length();
				if (hasDot && overMsLength >= 0) {
					return s.substring(0, s.length() - overMsLength);
				} else if (hasDot && lessMsLength >= 0) {
					for (int i = 0; i < lessMsLength; i++) {
						s += "0";
					}
					return s;
				}
				if (hasDot) {
					s += ".000";
				}
			}
			return s;
		}
	}

	public static Date currentData() {
		return new Date();
	}

	public static long getNowMilliSecondTime() {
		return currentData().getTime();
	}

	public static String getNowDateYYYYMM() {
		SimpleDateFormat formatter = new SimpleDateFormat("yyyyMM");
		return formatter.format(currentData());
	}

	public static String getNowDateYYYYMMDD() {
		SimpleDateFormat formatter = new SimpleDateFormat("yyyyMMdd");
		return formatter.format(currentData());
	}

	public static String getTheDayBeforNowDateYYYYMMDD() {
		SimpleDateFormat formatter = new SimpleDateFormat("yyyyMMdd");
		Calendar calendar = Calendar.getInstance();
		calendar.setTime(currentData());
		calendar.add(Calendar.DAY_OF_MONTH, -1);
		return formatter.format(calendar.getTime());
	}

	public static String getTheDayBeforAWeekYYYYMMDD() {
		SimpleDateFormat formatter = new SimpleDateFormat("yyyyMMdd");
		Calendar calendar = Calendar.getInstance();
		calendar.setTime(currentData());
		calendar.add(Calendar.DAY_OF_MONTH, -7);
		return formatter.format(calendar.getTime());
	}

	public static String getNowDateFormatCustom(String format) {
		SimpleDateFormat formatter = new SimpleDateFormat(format);
		return formatter.format(currentData());
	}

	public static long getNowDaySecondTime0() {
		Calendar c = Calendar.getInstance();
		Date d = currentData();
		c.setTime(d);
		c.set(c.get(Calendar.YEAR), c.get(Calendar.MONTH), c.get(Calendar.DAY_OF_MONTH), 0, 0, 0);
		return c.getTime().getTime() / 1000;
	}

	public static int getDayOfMonth() {
		Calendar now = Calendar.getInstance();
		return now.get(Calendar.DAY_OF_MONTH);
	}

	public static String toyyyyMMddHHmmss(long currentTime) {
		Date date = new Date(currentTime);
		SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddHHmmss");
		String timeStr = sdf.format(date);
		return timeStr;
	}

	public static String toyyyyMMddHHmmss(Date date) {
		SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddHHmmss");
		String timeStr = sdf.format(date);
		return timeStr;
	}

	public static Date getUtilDate(String date, String formatter) throws Exception {
		SimpleDateFormat sdf = new SimpleDateFormat(formatter);
		return sdf.parse(date);
	}

	public static long getOldTime(int unitCount, int unit) {
		Calendar c = Calendar.getInstance();
		c.add(unit, -unitCount);
		return c.getTimeInMillis();
	}

	public static Date toDate(String timeString) throws Exception {
		if (com.alibaba.druid.util.StringUtils.isEmpty(timeString)) {
			return null;
		}
		return DATE_FORMAT_ENUM.formatDate(timeString);
	}

	public static Date toDate(DateTime dateTime) {
		if (null == dateTime) {
			return null;
		}
		return dateTime.toDate();
	}

	public static Date toDate(Timestamp timestamp) {
		if (null == timestamp) {
			return null;
		}
		return new Date(timestamp.getTime());
	}

	public static DateTime toDateTime(Date date) {
		if (null == date) {
			return null;
		}
		return new DateTime(date);
	}

	public static DateTime toDateTime(Long timeLongInMicros) {
		if (null == timeLongInMicros || timeLongInMicros.longValue() == 0L) {
			return null;
		}
		return toDateTime(new Date(timeLongInMicros));
	}

	public static DateTime toDateTime(String timeString) throws Exception {
		if (com.alibaba.druid.util.StringUtils.isEmpty(timeString)) {
			return null;
		}
		return toDateTime(toDate(timeString));
	}

	public static Timestamp toTimestamp(Date date) {
		if (null == date) {
			return null;
		}
		return new Timestamp(date.getTime());
	}

	public static Timestamp toTimestamp(Long timeLongInMicros) {
		if (null == timeLongInMicros || timeLongInMicros.longValue() == 0L) {
			return null;
		}
		return toTimestamp(new Date(timeLongInMicros));
	}

	public static Timestamp toTimestamp(String timeString) throws Exception {
		if (com.alibaba.druid.util.StringUtils.isEmpty(timeString)) {
			return null;
		}
		return toTimestamp(toDate(timeString));
	}

	public static DateTime toDateTime(Timestamp timestamp) throws Exception {
		if (null == timestamp) {
			return null;
		}
		return toDateTime(toDate(timestamp));
	}

	public static Timestamp toTimestamp(DateTime dateTime) throws Exception {
		if (null == dateTime) {
			return null;
		}
		return toTimestamp(toDate(dateTime));
	}

	public static Long toLong(Date date) {
		if (null == date) {
			return null;
		}
		return date.getTime() * 1000;
	}

	public static Long toLong(String timeString) throws Exception {
		if (null == timeString) {
			return null;
		}
		return toLong(toDate(timeString));
	}

	public static Long toLong(DateTime dateTime) throws Exception {
		if (null == dateTime) {
			return null;
		}
		return toLong(toDate(dateTime));
	}

	public static Long toLong(Timestamp timestamp) throws Exception {
		if (null == timestamp) {
			return null;
		}
		return toLong(toDate(timestamp));
	}

	public static java.sql.Date toSqlDate(Date date) {
		if (null == date) {
			return null;
		}
		return new java.sql.Date(date.getTime());
	}

	public static java.sql.Date toSqlDate(DateTime dateTime) {
		if (null == dateTime) {
			return null;
		}
		return new java.sql.Date(toDate(dateTime).getTime());
	}

	public static java.sql.Date toSqlDate(Long timeLongInMicros) {
		if (null == timeLongInMicros || timeLongInMicros.longValue() == 0L) {
			return null;
		}
		return toSqlDate(new Date(timeLongInMicros));
	}
    // 指定模式的时间格式
    private static SimpleDateFormat getSDFormat(String pattern) {
        return new SimpleDateFormat(pattern);
    }

    /**
     * 当前日历，这里用中国时间表示
     *
     * @return 以当地时区表示的系统当前日历
     */
    public static Calendar getCalendar() {
        return Calendar.getInstance();
    }

    /**
     * 指定毫秒数表示的日历
     *
     * @param millis 毫秒数
     * @return 指定毫秒数表示的日历
     */
    public static Calendar getCalendar(long millis) {
        Calendar cal = Calendar.getInstance();
        // --------------------cal.setTimeInMillis(millis);
        cal.setTime(new Date(millis));
        return cal;
    }

    // ////////////////////////////////////////////////////////////////////////////
    // getDate
    // 各种方式获取的Date
    // ////////////////////////////////////////////////////////////////////////////

    /**
     * 当前日期
     *
     * @return 系统当前时间
     */
    public static Date getDate() {
        return new Date();
    }

    /**
     * 指定毫秒数表示的日期
     *
     * @param millis 毫秒数
     * @return 指定毫秒数表示的日期
     */
    public static Date getDate(long millis) {
        return new Date(millis);
    }

    /**
     * 时间戳转换为字符串
     *
     * @param time
     * @return
     */
    public static String timestamptoStr(Timestamp time) {
        Date date = null;
        if (null != time) {
            date = new Date(time.getTime());
        }
        return date2Str(date_sdf.get());
    }

    /**
     * 字符串转换时间戳
     *
     * @param str
     * @return
     */
    public static Timestamp str2Timestamp(String str) {
        Date date = str2Date(str, date_sdf.get());
        return new Timestamp(date.getTime());
    }

    /**
     * 字符串转换成日期
     *
     * @param str
     * @param sdf
     * @return
     */
    public static Date str2Date(String str, SimpleDateFormat sdf) {
        if (null == str || "".equals(str)) {
            return null;
        }
        Date date = null;
        try {
            date = sdf.parse(str);
            return date;
        } catch (ParseException e) {
            e.printStackTrace();
        }
        return null;
    }

    /**
     * 日期转换为字符串
     *
     * @param date_sdf 日期格式
     * @return 字符串
     */
    public static String date2Str(SimpleDateFormat date_sdf) {
        Date date = getDate();
        if (null == date) {
            return null;
        }
        return date_sdf.format(date);
    }

    /**
     * 格式化时间
     *
     * @param date
     * @param format
     * @return
     */
    public static String dateformat(String date, String format) {
        SimpleDateFormat sformat = new SimpleDateFormat(format);
        Date _date = null;
        try {
            _date = sformat.parse(date);
        } catch (ParseException e) {
            e.printStackTrace();
        }
        return sformat.format(_date);
    }

    /**
     * 日期转换为字符串
     *
     * @param date     日期
     * @param date_sdf 日期格式
     * @return 字符串
     */
    public static String date2Str(Date date, SimpleDateFormat date_sdf) {
        if (null == date) {
            return null;
        }
        return date_sdf.format(date);
    }

    /**
     * 日期转换为字符串
     *
     * @param format 日期格式
     * @return 字符串
     */
    public static String getDate(String format) {
        Date date = new Date();
        if (null == date) {
            return null;
        }
        SimpleDateFormat sdf = new SimpleDateFormat(format);
        return sdf.format(date);
    }

    /**
     * 指定毫秒数的时间戳
     *
     * @param millis 毫秒数
     * @return 指定毫秒数的时间戳
     */
    public static Timestamp getTimestamp(long millis) {
        return new Timestamp(millis);
    }

    /**
     * 以字符形式表示的时间戳
     *
     * @param time 毫秒数
     * @return 以字符形式表示的时间戳
     */
    public static Timestamp getTimestamp(String time) {
        return new Timestamp(Long.parseLong(time));
    }

    /**
     * 系统当前的时间戳
     *
     * @return 系统当前的时间戳
     */
    public static Timestamp getTimestamp() {
        return new Timestamp(System.currentTimeMillis());
    }

    /**
     * 当前时间，格式 yyyy-MM-dd HH:mm:ss
     *
     * @return 当前时间的标准形式字符串
     */
    public static String now() {
        return datetimeFormat.get().format(getCalendar().getTime());
    }

    /**
     * 指定日期的时间戳
     *
     * @param date 指定日期
     * @return 指定日期的时间戳
     */
    public static Timestamp getTimestamp(Date date) {
        return new Timestamp(date.getTime());
    }

    /**
     * 指定日历的时间戳
     *
     * @param cal 指定日历
     * @return 指定日历的时间戳
     */
    public static Timestamp getCalendarTimestamp(Calendar cal) {
        // ---------------------return new Timestamp(cal.getTimeInMillis());
        return new Timestamp(cal.getTime().getTime());
    }

    public static Timestamp gettimestamp() {
        Date dt = new Date();
        DateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        String nowTime = df.format(dt);
        java.sql.Timestamp buydate = java.sql.Timestamp.valueOf(nowTime);
        return buydate;
    }

    // ////////////////////////////////////////////////////////////////////////////
    // getMillis
    // 各种方式获取的Millis
    // ////////////////////////////////////////////////////////////////////////////

    /**
     * 系统时间的毫秒数
     *
     * @return 系统时间的毫秒数
     */
    public static long getMillis() {
        return System.currentTimeMillis();
    }

    /**
     * 指定日历的毫秒数
     *
     * @param cal 指定日历
     * @return 指定日历的毫秒数
     */
    public static long getMillis(Calendar cal) {
        // --------------------return cal.getTimeInMillis();
        return cal.getTime().getTime();
    }

    /**
     * 指定日期的毫秒数
     *
     * @param date 指定日期
     * @return 指定日期的毫秒数
     */
    public static long getMillis(Date date) {
        return date.getTime();
    }

    /**
     * 指定时间戳的毫秒数
     *
     * @param ts 指定时间戳
     * @return 指定时间戳的毫秒数
     */
    public static long getMillis(Timestamp ts) {
        return ts.getTime();
    }

    // ////////////////////////////////////////////////////////////////////////////
    // formatDate
    // 将日期按照一定的格式转化为字符串
    // ////////////////////////////////////////////////////////////////////////////

    /**
     * 默认方式表示的系统当前日期，具体格式：年-月-日
     *
     * @return 默认日期按“年-月-日“格式显示
     */
    public static String formatDate() {
        return date_sdf.get().format(getCalendar().getTime());
    }

    /**
     * 默认方式表示的系统当前日期，具体格式：yyyy-MM-dd HH:mm:ss
     *
     * @return 默认日期按“yyyy-MM-dd HH:mm:ss“格式显示
     */
    public static String formatDateTime() {
        return datetimeFormat.get().format(getCalendar().getTime());
    }

    /**
     * 获取时间字符串
     */
    public static String getDataString(SimpleDateFormat formatstr) {
        return formatstr.format(getCalendar().getTime());
    }

    /**
     * 指定日期的默认显示，具体格式：年-月-日
     *
     * @param cal 指定的日期
     * @return 指定日期按“年-月-日“格式显示
     */
    public static String formatDate(Calendar cal) {
        return date_sdf.get().format(cal.getTime());
    }

    /**
     * 指定日期的默认显示，具体格式：年-月-日
     *
     * @param date 指定的日期
     * @return 指定日期按“年-月-日“格式显示
     */
    public static String formatDate(Date date) {
        return date_sdf.get().format(date);
    }

    /**
     * 指定毫秒数表示日期的默认显示，具体格式：年-月-日
     *
     * @param millis 指定的毫秒数
     * @return 指定毫秒数表示日期按“年-月-日“格式显示
     */
    public static String formatDate(long millis) {
        return date_sdf.get().format(new Date(millis));
    }

    /**
     * 默认日期按指定格式显示
     *
     * @param pattern 指定的格式
     * @return 默认日期按指定格式显示
     */
    public static String formatDate(String pattern) {
        return getSDFormat(pattern).format(getCalendar().getTime());
    }

    /**
     * 指定日期按指定格式显示
     *
     * @param cal     指定的日期
     * @param pattern 指定的格式
     * @return 指定日期按指定格式显示
     */
    public static String formatDate(Calendar cal, String pattern) {
        return getSDFormat(pattern).format(cal.getTime());
    }

    /**
     * 指定日期按指定格式显示
     *
     * @param date    指定的日期
     * @param pattern 指定的格式
     * @return 指定日期按指定格式显示
     */
    public static String formatDate(Date date, String pattern) {
        return getSDFormat(pattern).format(date);
    }

    // ////////////////////////////////////////////////////////////////////////////
    // formatTime
    // 将日期按照一定的格式转化为字符串
    // ////////////////////////////////////////////////////////////////////////////

    /**
     * 默认方式表示的系统当前日期，具体格式：年-月-日 时：分
     *
     * @return 默认日期按“年-月-日 时：分“格式显示
     */
    public static String formatTime() {
        return time_sdf.get().format(getCalendar().getTime());
    }

    /**
     * 指定毫秒数表示日期的默认显示，具体格式：年-月-日 时：分
     *
     * @param millis 指定的毫秒数
     * @return 指定毫秒数表示日期按“年-月-日 时：分“格式显示
     */
    public static String formatTime(long millis) {
        return time_sdf.get().format(new Date(millis));
    }

    /**
     * 指定日期的默认显示，具体格式：年-月-日 时：分
     *
     * @param cal 指定的日期
     * @return 指定日期按“年-月-日 时：分“格式显示
     */
    public static String formatTime(Calendar cal) {
        return time_sdf.get().format(cal.getTime());
    }

    /**
     * 指定日期的默认显示，具体格式：年-月-日 时：分
     *
     * @param date 指定的日期
     * @return 指定日期按“年-月-日 时：分“格式显示
     */
    public static String formatTime(Date date) {
        return time_sdf.get().format(date);
    }

    // ////////////////////////////////////////////////////////////////////////////
    // formatShortTime
    // 将日期按照一定的格式转化为字符串
    // ////////////////////////////////////////////////////////////////////////////

    /**
     * 默认方式表示的系统当前日期，具体格式：时：分
     *
     * @return 默认日期按“时：分“格式显示
     */
    public static String formatShortTime() {
        return short_time_sdf.get().format(getCalendar().getTime());
    }

    /**
     * 指定毫秒数表示日期的默认显示，具体格式：时：分
     *
     * @param millis 指定的毫秒数
     * @return 指定毫秒数表示日期按“时：分“格式显示
     */
    public static String formatShortTime(long millis) {
        return short_time_sdf.get().format(new Date(millis));
    }

    /**
     * 指定日期的默认显示，具体格式：时：分
     *
     * @param cal 指定的日期
     * @return 指定日期按“时：分“格式显示
     */
    public static String formatShortTime(Calendar cal) {
        return short_time_sdf.get().format(cal.getTime());
    }

    /**
     * 指定日期的默认显示，具体格式：时：分
     *
     * @param date 指定的日期
     * @return 指定日期按“时：分“格式显示
     */
    public static String formatShortTime(Date date) {
        return short_time_sdf.get().format(date);
    }

    // ////////////////////////////////////////////////////////////////////////////
    // parseDate
    // parseCalendar
    // parseTimestamp
    // 将字符串按照一定的格式转化为日期或时间
    // ////////////////////////////////////////////////////////////////////////////

    /**
     * 根据指定的格式将字符串转换成Date 如输入：2003-11-19 11:20:20将按照这个转成时间
     *
     * @param src     将要转换的原始字符窜
     * @param pattern 转换的匹配格式
     * @return 如果转换成功则返回转换后的日期
     * @throws ParseException
     */
    public static Date parseDate(String src, String pattern) throws ParseException {
        return getSDFormat(pattern).parse(src);

    }

    /**
     * 根据指定的格式将字符串转换成Date 如输入：2003-11-19 11:20:20将按照这个转成时间
     *
     * @param src     将要转换的原始字符窜
     * @param pattern 转换的匹配格式
     * @return 如果转换成功则返回转换后的日期
     * @throws ParseException
     */
    public static Calendar parseCalendar(String src, String pattern) throws ParseException {

        Date date = parseDate(src, pattern);
        Calendar cal = Calendar.getInstance();
        cal.setTime(date);
        return cal;
    }

    public static String formatAddDate(String src, String pattern, int amount) throws ParseException {
        Calendar cal;
        cal = parseCalendar(src, pattern);
        cal.add(Calendar.DATE, amount);
        return formatDate(cal);
    }

    /**
     * 根据指定的格式将字符串转换成Date 如输入：2003-11-19 11:20:20将按照这个转成时间
     *
     * @param src     将要转换的原始字符窜
     * @param pattern 转换的匹配格式
     * @return 如果转换成功则返回转换后的时间戳
     * @throws ParseException
     */
    public static Timestamp parseTimestamp(String src, String pattern) throws ParseException {
        Date date = parseDate(src, pattern);
        return new Timestamp(date.getTime());
    }

    // ////////////////////////////////////////////////////////////////////////////
    // dateDiff
    // 计算两个日期之间的差值
    // ////////////////////////////////////////////////////////////////////////////

    /**
     * 计算两个时间之间的差值，根据标志的不同而不同
     *
     * @param flag   计算标志，表示按照年/月/日/时/分/秒等计算
     * @param calSrc 减数
     * @param calDes 被减数
     * @return 两个日期之间的差值
     */
    public static int dateDiff(char flag, Calendar calSrc, Calendar calDes) {

        long millisDiff = getMillis(calSrc) - getMillis(calDes);

        if (flag == 'y') {
            return (calSrc.get(Calendar.YEAR) - calDes.get(Calendar.YEAR));
        }

        if (flag == 'd') {
            return (int) (millisDiff / DAY_IN_MILLIS);
        }

        if (flag == 'h') {
            return (int) (millisDiff / HOUR_IN_MILLIS);
        }

        if (flag == 'm') {
            return (int) (millisDiff / MINUTE_IN_MILLIS);
        }

        if (flag == 's') {
            return (int) (millisDiff / SECOND_IN_MILLIS);
        }

        return 0;
    }

    /**
     * String类型 转换为Date, 如果参数长度为10 转换格式”yyyy-MM-dd“ 如果参数长度为19 转换格式”yyyy-MM-dd
     * HH:mm:ss“ * @param text String类型的时间值
     */
    @Override
    public void setAsText(String text) throws IllegalArgumentException {
        if (StringUtils.hasText(text)) {
            try {
                if (text.indexOf(":") == -1 && text.length() == 10) {
                    setValue(DateUtils.date_sdf.get().parse(text));
                } else if (text.indexOf(":") > 0 && text.length() == 19) {
                    setValue(DateUtils.datetimeFormat.get().parse(text));
                } else {
                    throw new IllegalArgumentException("Could not parse date, date format is error ");
                }
            } catch (ParseException ex) {
                IllegalArgumentException iae = new IllegalArgumentException("Could not parse date: " + ex.getMessage());
                iae.initCause(ex);
                throw iae;
            }
        } else {
            setValue(null);
        }
    }

    public static int getYear() {
        GregorianCalendar calendar = new GregorianCalendar();
        calendar.setTime(getDate());
        return calendar.get(Calendar.YEAR);
    }

}
